home *** CD-ROM | disk | FTP | other *** search
/ Cream of the Crop 21 / Cream of the Crop 21 (Terry Blount) (October 1996).iso / os2 / e33el2.zip / emacs / 19.33 / lisp / derived.el < prev    next >
Lisp/Scheme  |  1996-05-27  |  14KB  |  353 lines

  1. ;;; derived.el --- allow inheritance of major modes.
  2. ;;; (formerly mode-clone.el)
  3.  
  4. ;; Copyright (C) 1993, 1994 Free Software Foundation, Inc.
  5.  
  6. ;; Author: David Megginson (dmeggins@aix1.uottawa.ca)
  7. ;; Maintainer: FSF
  8.  
  9. ;; This file is part of GNU Emacs.
  10.  
  11. ;; GNU Emacs is free software; you can redistribute it and/or modify
  12. ;; it under the terms of the GNU General Public License as published by
  13. ;; the Free Software Foundation; either version 2, or (at your option)
  14. ;; any later version.
  15.  
  16. ;; GNU Emacs is distributed in the hope that it will be useful,
  17. ;; but WITHOUT ANY WARRANTY; without even the implied warranty of
  18. ;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
  19. ;; GNU General Public License for more details.
  20.  
  21. ;; You should have received a copy of the GNU General Public License
  22. ;; along with GNU Emacs; see the file COPYING.  If not, write to the
  23. ;; Free Software Foundation, Inc., 59 Temple Place - Suite 330,
  24. ;; Boston, MA 02111-1307, USA.
  25.  
  26. ;;; Commentary:
  27.  
  28. ;; GNU Emacs is already, in a sense, object oriented -- each object
  29. ;; (buffer) belongs to a class (major mode), and that class defines
  30. ;; the relationship between messages (input events) and methods
  31. ;; (commands) by means of a keymap.
  32. ;;
  33. ;; The only thing missing is a good scheme of inheritance.  It is
  34. ;; possible to simulate a single level of inheritance with generous
  35. ;; use of hooks and a bit of work -- sgml-mode, for example, also runs
  36. ;; the hooks for text-mode, and keymaps can inherit from other keymaps
  37. ;; -- but generally, each major mode ends up reinventing the wheel.
  38. ;; Ideally, someone should redesign all of Emacs's major modes to
  39. ;; follow a more conventional object-oriented system: when defining a
  40. ;; new major mode, the user should need only to name the existing mode
  41. ;; it is most similar to, then list the (few) differences.
  42. ;;
  43. ;; In the mean time, this package offers most of the advantages of
  44. ;; full inheritance with the existing major modes.  The macro
  45. ;; `define-derived-mode' allows the user to make a variant of an existing
  46. ;; major mode, with its own keymap.  The new mode will inherit the key
  47. ;; bindings of its parent, and will, in fact, run its parent first
  48. ;; every time it is called.  For example, the commands
  49. ;;
  50. ;;  (define-derived-mode hypertext-mode text-mode "Hypertext"
  51. ;;    "Major mode for hypertext.\n\n\\{hypertext-mode-map}"
  52. ;;    (setq case-fold-search nil))
  53. ;;
  54. ;;  (define-key hypertext-mode-map [down-mouse-3] 'do-hyper-link)
  55. ;;
  56. ;; will create a function `hypertext-mode' with its own (sparse)
  57. ;; keymap `hypertext-mode-map.'  The command M-x hypertext-mode will
  58. ;; perform the following actions:
  59. ;;
  60. ;; - run the command (text-mode) to get its default setup
  61. ;; - replace the current keymap with 'hypertext-mode-map,' which will
  62. ;;   inherit from 'text-mode-map'.
  63. ;; - replace the current syntax table with
  64. ;;   'hypertext-mode-syntax-table', which will borrow its defaults
  65. ;;   from the current text-mode-syntax-table.
  66. ;; - replace the current abbrev table with
  67. ;;   'hypertext-mode-abbrev-table', which will borrow its defaults
  68. ;;   from the current text-mode-abbrev table
  69. ;; - change the mode line to read "Hypertext"
  70. ;; - assign the value 'hypertext-mode' to the 'major-mode' variable
  71. ;; - run the body of commands provided in the macro -- in this case,
  72. ;;   set the local variable `case-fold-search' to nil.
  73. ;; - **run the command (hypertext-mode-setup), which is empty by
  74. ;;   default, but may be redefined by the user to contain special
  75. ;;   commands (ie. setting local variables like 'outline-regexp')
  76. ;;   **NOTE: do not use this option -- it will soon be obsolete.
  77. ;; - run anything assigned to 'hypertext-mode-hooks' (obsolete, but
  78. ;;   supported for the sake of compatibility).
  79. ;;
  80. ;; The advantages of this system are threefold.  First, text mode is
  81. ;; untouched -- if you had added the new keystroke to `text-mode-map,'
  82. ;; possibly using hooks, you would have added it to all text buffers
  83. ;; -- here, it appears only in hypertext buffers, where it makes
  84. ;; sense.  Second, it is possible to build even further, and make
  85. ;; a derived mode from a derived mode.  The commands
  86. ;;
  87. ;;   (define-derived-mode html-mode hypertext-mode "HTML")
  88. ;;   [various key definitions]
  89. ;; 
  90. ;; will add a new major mode for HTML with very little fuss.
  91. ;;
  92. ;; Note also the function `derived-mode-class,' which returns the non-derived
  93. ;; major mode which a derived mode is based on (ie. NOT necessarily the
  94. ;; immediate parent).
  95. ;;
  96. ;; (derived-mode-class 'text-mode) ==> text-mode
  97. ;; (derived-mode-class 'hypertext-mode) ==> text-mode
  98. ;; (derived-mode-class 'html-mode) ==> text-mode
  99.  
  100. ;;; Code:
  101.  
  102. ;; PUBLIC: define a new major mode which inherits from an existing one.
  103.  
  104. ;;;###autoload
  105. (defmacro define-derived-mode (child parent name &optional docstring &rest body)
  106.   "Create a new mode as a variant of an existing mode.
  107.  
  108. The arguments to this command are as follow:
  109.  
  110. CHILD:     the name of the command for the derived mode.
  111. PARENT:    the name of the command for the parent mode (ie. text-mode).
  112. NAME:      a string which will appear in the status line (ie. \"Hypertext\")
  113. DOCSTRING: an optional documentation string--if you do not supply one,
  114.            the function will attempt to invent something useful.
  115. BODY:      forms to execute just before running the
  116.            hooks for the new mode.
  117.  
  118. Here is how you could define LaTeX-Thesis mode as a variant of LaTeX mode:
  119.  
  120.   (define-derived-mode LaTeX-thesis-mode LaTeX-mode \"LaTeX-Thesis\")
  121.  
  122. You could then make new key bindings for `LaTeX-thesis-mode-map'
  123. without changing regular LaTeX mode.  In this example, BODY is empty,
  124. and DOCSTRING is generated by default.
  125.  
  126. On a more complicated level, the following command uses sgml-mode as
  127. the parent, and then sets the variable `case-fold-search' to nil:
  128.  
  129.   (define-derived-mode article-mode sgml-mode \"Article\"
  130.     \"Major mode for editing technical articles.\"
  131.     (setq case-fold-search nil))
  132.  
  133. Note that if the documentation string had been left out, it would have
  134. been generated automatically, with a reference to the keymap."
  135.  
  136.                     ; Some trickiness, since what
  137.                     ; appears to be the docstring
  138.                     ; may really be the first
  139.                     ; element of the body.
  140.   (if (and docstring (not (stringp docstring)))
  141.       (progn (setq body (cons docstring body))
  142.          (setq docstring nil)))
  143.   (setq docstring (or docstring (derived-mode-make-docstring parent child)))
  144.  
  145.   (` (progn 
  146.        (derived-mode-init-mode-variables (quote (, child)))
  147.        (defun (, child) ()
  148.      (, docstring)
  149.      (interactive)
  150.                     ; Run the parent.
  151.      ((, parent))
  152.                     ; Identify special modes.
  153.      (if (get (quote (, parent)) 'special)
  154.          (put (quote (, child)) 'special t))
  155.                     ; Identify the child mode.
  156.      (setq major-mode (quote (, child)))
  157.      (setq mode-name (, name))
  158.                     ; Set up maps and tables.
  159.      (derived-mode-set-keymap (quote (, child)))
  160.      (derived-mode-set-syntax-table (quote (, child)))
  161.      (derived-mode-set-abbrev-table (quote (, child)))
  162.                     ; Splice in the body (if any).
  163.      (,@ body)
  164. ;;;                    ; Run the setup function, if
  165. ;;;                    ; any -- this will soon be
  166. ;;;                    ; obsolete.
  167. ;;;     (derived-mode-run-setup-function (quote (, child)))
  168.                     ; Run the hooks, if any.
  169.      (derived-mode-run-hooks (quote (, child)))))))
  170.  
  171.  
  172. ;; PUBLIC: find the ultimate class of a derived mode.
  173.  
  174. (defun derived-mode-class (mode)
  175.   "Find the class of a major mode.
  176. A mode's class is the first ancestor which is NOT a derived mode.
  177. Use the `derived-mode-parent' property of the symbol to trace backwards."
  178.   (while (get mode 'derived-mode-parent)
  179.     (setq mode (get mode 'derived-mode-parent)))
  180.   mode)
  181.  
  182.  
  183. ;; Inline functions to construct various names from a mode name.
  184.  
  185. (defsubst derived-mode-setup-function-name (mode)
  186.   "Construct a setup-function name based on a mode name."
  187.   (intern (concat (symbol-name mode) "-setup")))
  188.  
  189. (defsubst derived-mode-hooks-name (mode)
  190.   "Construct a hooks name based on a mode name."
  191.   (intern (concat (symbol-name mode) "-hooks")))
  192.  
  193. (defsubst derived-mode-map-name (mode)
  194.   "